<?php

namespace app\models\tableModel;

use app\entity\PageListEntity;
use app\models\tableModel\base\User;
use app\service\tool\GenerateService;
use app\service\tool\UtilsService;
use yii\db\Expression;
use yii\db\ExpressionInterface;
use yii\helpers\ArrayHelper;

/**
 * [用户]数据库操作模型
 * 作者: Editor Name
 * 日期: 2022/01/07
 * 时间: 20:15:26
 */
class UserModel extends User
{

    /**
     * 性别 列表
     * @var array
     */
    private static $sexList = [
        [
            'key'   => 'boy',
            'value' => 1,
            'text'  => '男'
        ],
        [
            'key'   => 'girl',
            'value' => 2,
            'text'  => '女'
        ]
    ];

    /**
     * 状态 列表
     * @var array
     */
    private static $statusList = [
        [
            'key'   => 'disabled',
            'value' => -1,
            'text'  => '禁用'
        ],
        [
            'key'   => 'open',
            'value' => 1,
            'text'  => '开启'
        ]
    ];


    /**
     * 规则验证
     * @return array
     */
    public function rules()
    {

        // [性别]列表
        $sexList = array_column(self::getSexList(), 'value');
        // [状态]列表
        $statusList = array_column(self::getStatusList(), 'value');

        $parent = parent::rules();
        return ArrayHelper::merge($parent, [
            ['sex', 'in', 'range' => $sexList, 'message' => '性别不合法'],
            ['status', 'in', 'range' => $statusList, 'message' => '状态不合法'],
            [['user_name', 'phone'], 'unique']
        ]);
    }

    /**
     * 重写label的 文字
     */
    public function attributeLabels()
    {

        $parent = parent::attributeLabels();
        return array_merge($parent, [
            'id'          => '',
            'user_name'   => '账号',
            'nick_name'   => '昵称',
            'password'    => '密码',
            'salt'        => '加密字符',
            'sex'         => '性别',
            'avatar'      => '头像',
            'phone'       => '电话号码',
            'description' => '个人说明',
            'add_time'    => '注册时间',
            'update_time' => '更新时间',
            'reg_ip'      => '注册IP',
            'status'      => '状态',
        ]);
    }

    /**
     * 场景
     * @return array
     */
    public function scenarios()
    {

        $scenarios = parent::scenarios();
        return ArrayHelper::merge($scenarios, [
            // 自定义场景 (无用请删除)
            'scUpdate' => [
                'someAttributes'
            ]
        ]);
    }

    /**
     * 自定义过滤单条[where]
     *  ` 仅限于通过方法[loadWhere]传入的条件, 可以在其中自定义逻辑
     *  ` 保证最终返回值为一维数组[Yii2]通用条件格式[运算符, 字段名, 值]
     * @param $condition array [sql]查询条件, 严格遵循Yii多条件查询:[运算符, 字段名, 值]
     * @return array
     */
    public function whereCustomFilter($condition = [])
    {

        // // 例：字段为编号 —— 操作符使用[IN]
        // if ($field == 'id') {
        //     return ['IN', $condition[1], $value[2]];
        // }

        // 最终返回必须为数组：[运算符, 字段名, 值]，符合Yii2规范
        return $condition;
    }

    /**
     * 获取分页信息
     * @param integer $page 当前页
     * @param integer $limit 获取几条
     * @param null $field 获取字段
     * @return PageListEntity
     */
    public function getPaginate($page, $limit, $field = null)
    {

        // 当前页面计算
        $page = max($page - 1, 0);

        // 查找的 字段空的 就默认给列表
        if (!$field) $field = '*';

        // 基础[where]加载完毕
        $this->getBaseSql()->select($field);

        // 数据获取 分页等
        $list = $this->getBaseSql()->offset($page * $limit)
            ->limit($limit)
            ->asArray()->all();

        // 格式化数据后的列表
        $list = $this->formatData($list, $field);
        // 总数量
        $total = $this->getBaseSql()->count();

        return PageListEntity::loadModel(compact('list', 'total'));
    }

    /**
     * 格式化列表活详情数据
     * @param array $list 列表
     * @param array $opt 其他设置
     * @return mixed
     */
    public static function formatData($list, $opt = [])
    {

        // 为空直接返回
        if (empty($list)) return $list;
        // 需要返回第一组（可能不是二维数组）
        $needFirst = false;
        if (!is_array(array_values($list)[0])) {
            $needFirst = true;
            $list      = [$list];
        }

        ### 某些参数初始化

        ### 开始格式化
        foreach ($list as $k => &$v) {

            // 更新时间
            if (isset($v['add_time'])) {
                $v['add_time_text']   = date('Y-m-d H:i:s', $v['add_time']);
                $v['add_time_text_s'] = date('Y-m-d', $v['add_time']);
            }

            // 更新时间
            if (isset($v['update_time'])) {
                $v['update_time_text']   = date('Y-m-d H:i:s', $v['update_time']);
                $v['update_time_text_s'] = date('Y-m-d', $v['update_time']);
            }

            // 性别 文本
            if (isset($v['sex'])) {
                $v['sex_text'] = self::getSexText($v['sex']);
            }

            // 状态 文本
            if (isset($v['status'])) {
                $v['status_text'] = self::getStatusText($v['status']);
            }
        }

        reset($list);
        return $needFirst ? current($list) : $list;
    }

    /**
     * 添加|保存
     * @param bool $runValidation 是否仅仅运行验证，不运行保存
     * @param string[]|null|string $attributeNames 仅仅保存某几个字段
     * @return bool
     */
    public function save($runValidation = false, $attributeNames = null)
    {

        $nowTime = time();
        // 添加时间
        if (empty($this->add_time)) $this->add_time = $nowTime;
        // 更新时间
        $this->update_time = $nowTime;
        // 检测
        if ($this->hasErrors() || !$this->validate($attributeNames)) {

            // 记录下错误日志
            \Yii::error([
                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[用户]数据库操作模型]验证数据失败             ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $this->getAttributes(),
                $this->getErrors()
            ], 'db');
            return false;
        }

        // 需要 && 执行保存
        if (!$runValidation && !parent::save(false, $attributeNames)) {

            // 记录下错误日志
            \Yii::error([
                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[用户]数据库操作模型]保存数据失败             ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $this->getAttributes(),
                $this->getErrors()
            ], 'db');
            return false;
        }

        return true;
    }

    /**
     * [静态方法]批量快速更新某些字段
     * @param $condition
     * @param array $fieldVal
     * @return bool
     */
    public static function updateField($condition, $fieldVal = [])
    {

        // 验证字段
        $model = self::loadModel();
        $model->load($fieldVal, '');
        if (!$model->validate(array_keys($fieldVal))) {
            $error                          = UtilsService::getModelError($model->getErrors());
            self::$error_[$error['column']] = [$error['msg']];
            return false;
        }
        // 重新取值规则化后的特定值列表
        $fieldVal = $model->getAttributes(array_keys($fieldVal));

        $db = \Yii::$app->db->createCommand();

        try {

            $db->update(self::tableName(), $fieldVal, $condition)->execute();

            // 否则成功
            return true;
        } catch (\Exception $error) {

            // 记录下错误日志
            \Yii::error([
                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[用户]数据库操作模型]批量修改[指定字段]失败，   ``",
                "``         {$error->getMessage()}                       ``",
                "`` SQL语句: {$db->getRawSql()}                         ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $error->getTraceAsString()
            ], 'db');

            // 静态错误
            self::$error_['db_error'] = empty(self::$error_['db_error']) ?
                [$error->getMessage()] : array_merge(self::$error_['db_error'], [$error->getMessage()]);

            return false;
        }
    }

    /**
     * 批量添加数据
     * @param array $createData
     * @return bool
     */
    public static function createData($createData = [])
    {

        $db = \Yii::$app->db->createCommand();
        try {

            ### 先数据格式化
            // 如果是一维数组则二维数组化
            if (ArrayHelper::isAssociative($createData)) {
                $createData = [$createData];
            }
            $values = [];
            foreach ($createData as $k => $v) {

                $model = self::loadModel();
                $model->load($createData[$k], '');
                if (!$model->save(true)) {
                    // 取出错误信息
                    $error = UtilsService::getModelError($model->errors);
                    // 添加到静态方法上
                    self::$error_[$error['column']] = [$error['msg']];
                    return false;
                }

                $createData[$k] = $model->getAttributes($model::getTableSchema()->getColumnNames());

                // 循环一些数据
                foreach ($createData[$k] as $kc => $vc) {
                    // 字段类型为[JSON]类型需要转为[JSON]
                    if (is_array($vc)) {
                        $createData[$k][$kc] = json_encode($vc, JSON_UNESCAPED_UNICODE);
                        continue;
                    }
                }

                // 值赋值
                $values[] = array_values($createData[$k]);;
            }

            ### 取出此次操作的字段列表
            $columns = !current($createData) ? [] : array_keys(current($createData));

            // 执行
            $addResult = $db->batchInsert(self::tableName(), $columns, $values)->execute();

            return $addResult;
        } catch (\Exception $error) {

            // 记录下错误日志
            \Yii::error([

                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[用户]数据库操作模型]批量添加[数据]失败，      ``",
                "``         {$error->getMessage()}                       ``",
                "`` SQL语句: {$db->getRawSql()}                         ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $error->getTraceAsString()
            ], 'db');

            // 静态错误
            self::$error_['db_error'] = empty(self::$error_['db_error']) ?
                [$error->getMessage()] : array_merge(self::$error_['db_error'], [$error->getMessage()]);

            return false;
        }
    }

    /**
     * 更新某些字段自增|自减
     * @param $condition
     * @param array $fieldVal 增/减加的字段
     * @return bool
     */
    public static function updateCounter($condition, $fieldVal = [])
    {

        $model = new self();
        foreach ($fieldVal as $k => $v) {

            if (!$model->hasAttribute($k)) {

                unset($fieldVal[$k]);
                continue;
            }
        }

        try {

            $model->updateAllCounters($fieldVal, $condition);

            // 否则成功
            return true;
        } catch (\Exception $error) {

            // 记录下错误日志
            \Yii::error([

                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[用户]数据库操作模型]批量增/减[指定字段]失败   ``",
                "``         {$error->getMessage()}                       ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $error->getTraceAsString()
            ], 'db');

            // 静态错误
            self::$error_['db_error'] = empty(self::$error_['db_error']) ?
                [$error->getMessage()] : array_merge(self::$error_['db_error'], [$error->getMessage()]);

            return false;
        }
    }


    /**
     * 获取[性别][男]
     * @return mixed
     */
    public static function getSexBoy()
    {

        $list = array_column(self::$sexList, null, 'key');
        return $list['boy']['value'];
    }

    /**
     * 获取[性别]文本
     * @param $value
     * @return mixed|string
     */
    public static function getSexText($value)
    {

        // 列表
        $list = array_column(self::$sexList, null, 'value');
        // 不合法 - 不存在
        if (empty($list[$value]['text'])) return '--';

        // 最终正常返回
        return $list[$value]['text'];
    }

    /**
     * 获取[性别]列表 值
     * @return mixed|string
     */
    public static function getSexList()
    {

        // 最终正常返回
        return self::$sexList;
    }


    /**
     * 获取[状态][关闭]值
     * @return mixed
     */
    public static function getStatusDisabled()
    {

        $list = array_column(self::$statusList, null, 'key');
        return $list['disabled']['value'];
    }

    /**
     * 获取[状态][开启]值
     * @return mixed
     */
    public static function getStatusOpen()
    {

        $list = array_column(self::$statusList, null, 'key');
        return $list['open']['value'];
    }

    /**
     * 获取[状态]文本
     * @param $value
     * @return mixed|string
     */
    public static function getStatusText($value)
    {

        // 列表
        $list = array_column(self::$statusList, null, 'value');
        // 不合法 - 不存在
        if (empty($list[$value]['text'])) return '--';

        // 最终正常返回
        return $list[$value]['text'];
    }

    /**
     * 获取[状态]列表 值
     * @return mixed|string
     */
    public static function getStatusList()
    {

        // 最终正常返回
        return self::$statusList;
    }

    /**
     * 获取静态错误
     * @return mixed
     */
    public static function getStaticErrors()
    {
        return self::$error_;
    }


    /**
     * 密码密文获取
     * @param $password
     * @param $salt
     * @return string
     */
    protected function getPwdCrypt($password, $salt)
    {
        return md5($password . $salt);
    }

    /**
     * 验证密码是否匹配
     * @param $password
     * @return bool
     */
    public function passwordCanLogin($password)
    {
        // 现在密码加密
        $nowPassword = $this->getPwdCrypt($password, $this->salt);
        // 和传输的密码比对
        if ($nowPassword != $this->password) return false;

        return true;
    }

    /**
     * 密码生成为密文
     * @param string $newPassword 新的密码，不传输将以成员变量password为明文密码操作
     * @param bool $reloadSalt 是否强制生成密码随机字符串
     * @return bool|string
     */
    public function passwordCrypt($newPassword, $reloadSalt = false)
    {

        ### 密码验证
        if (empty($newPassword)) {
            $this->addError('password', '密码不能为空');
            return false;
        }
        // 长度
        if (!preg_match('/^[A-Za-z_0-9]{6,16}$/', $newPassword)) {
            $this->addError('password', '密码长度为9-16');
            return false;
        }
        // 内容
        if (!!preg_match('/^(?![0-9]+$)(?![a-zA-Z]+$)[A-Za-z_0-9]{6,16}$/', $newPassword)) {
            $this->addError('password', '密码必须是数字字母组合');
            return false;
        }

        // 随机串空的生成随机串
        if (empty($this->salt) || $reloadSalt) $this->getNewSalt();
        $this->setAttribute('password', $this->getPwdCrypt($newPassword, $this->salt));

        return $this->getAttribute('password');
    }

    /**
     * 生成密码[SALT]随机加密字符
     * @return string
     */
    public function getNewSalt()
    {
        $this->salt = GenerateService::getRandomStr(8);
        return $this->salt;
    }
}
